Konwersja MBR <=> GPT
Kategoria: Artykuły, etykiety: mbr, konwersja, gpt, dysk
Dodany: 2014-01-04 16:54
(zmodyfikowany: 2014-01-11 01:52)
Przez: morfik
Wyświetleń: 24607
Od jakiegoś czasu nosiłem się z zamiarem utworzenia na swoim dysku tablicy partycji GPT. Problem w tym, że tam jest wgranych paręset GiB danych i nie mam gdzie tego upchnąć. Jedyne co mi wpadło do głowy to przekonwertowanie tablicy MBR na GPT. Nie żebym jakoś tego potrzebował, w końcu mam dysk 1,5TB ale tak z ciekawości chciałem zobaczyć czy da się to zrobić w sposób łatwy i bezproblemowy. No i dało się. W roli głównej wystąpił mój stary dysk (80GB) -- zrobiłem na nim 6 partycji, 3 zwykłe i 3 rozszerzone. Pierwsza partycja jest pod wina xp, druga to zaszyfrowany system linuxowy ale jak zwykle zapomniałem zostawić trochę miejsca na partycję /boot , a ta wylądowała na pozycji sda6. Partycja trzecia to niezaszyfrowany swap, a pozostałe dwie (5 i 7) to dwa zaszyfrowane voluminy, z których jeden celowo umieszczony na końcu dysku, drugi zaś na początku rozszerzonej partycji. Cel to przekonwertować tablicę partycji MBR na GPT (i vice versa) bez utraty danych.
Konwersja MBR -> GPT
Tak wygląda dysk widziany oczami parted (z uwzględnieniem wolnych przestrzeni):
(parted) print free
Model: ATA WDC WD800JB-00JJ (scsi)
Disk /dev/sda: 80,0GB
Sector size (logical/physical): 512B/512B
Partition Table: msdos
Number Start End Size Type File system Flags
32,3kB 1049kB 1016kB Free Space
1 1049kB 31,5GB 31,5GB primary ntfs boot
2 31,5GB 43,0GB 11,5GB primary
3 43,0GB 44,1GB 1074MB primary linux-swap(v1)
4 44,1GB 80,0GB 36,0GB extended
5 44,1GB 54,6GB 10,5GB logical
6 54,6GB 65,0GB 10,5GB logical ext4 boot
7 65,0GB 80,0GB 15,0GB logical
80,0GB 80,0GB 56,8kB Free Space
By przeprowadzić bezproblemowo konwersję z MBR na GPT trzeba mieć troszeczkę wolnego miejsca na początku i na końcu dysku. Ile? Struktura GPT wygląda tak: 512 bajtowy MBR + 512bajtowy nagłówek GPT + 16 KiB głównej tablicy partycji. Łącznie daje to 34 sektory 512 bajtowe, co przekłada się na 34*512=17408 bajtów wolnego miejsca na początku dysku. Ale tablica partycji GPT ma jeszcze backup swojej zawartości, który jest zlokalizowany na końcu dysku i w jego skład wchodzi kopia głównej tablicy partycji -- 16 KiB oraz kopia nagłówka 1 sektor 512 bajtowy. Łącznie 33 sektory co daje 16896 bajtów potrzebnego miejsca na końcu dysku.
Obecnie większość narzędzi tworzących partycje na dysku ustawia równanie do 1MiB co przekłada się na 2048 wolnych sektorów na początku dysku, więc tutaj nie powinno być problemów ze spełnieniem wymagań pod GPT. Ale takie równanie do 1MiB może czasem zając resztę dysku. W moim przypadku, jak widać powyżej, zostało trochę wolnego miejsca -- 56,8kB. Także u mnie wszystko jest ok. W przypadku jednak gdyby partycja wypełniła dysk do końca, trzeba będzie ją skrócić.
By przekonwertować tablicę partycji odpalamy gdisk:
# gdisk /dev/sda
GPT fdisk (gdisk) version 0.8.5
Partition table scan:
MBR: MBR only
BSD: not present
APM: not present
GPT: not present
***************************************************************
Found invalid GPT and valid MBR; converting MBR to GPT format.
THIS OPERATION IS POTENTIALLY DESTRUCTIVE! Exit by typing 'q' if
you don't want to convert your MBR partitions to GPT format!
***************************************************************
Jak widać, gdisk wykrył, że ma do czynienia z tablicą partycji MBR. Konwertowanie polega na wpisaniu w:
Command (? for help): w
Final checks complete. About to write GPT data. THIS WILL OVERWRITE EXISTING
PARTITIONS!!
Do you want to proceed? (Y/N): y
OK; writing new GUID partition table (GPT) to /dev/sda.
The operation has completed successfully.
Sprawdzamy, czy faktycznie się przekonwertowało:
# gdisk /dev/sda
GPT fdisk (gdisk) version 0.8.5
Partition table scan:
MBR: protective
BSD: not present
APM: not present
GPT: present
Found valid GPT with protective MBR; using GPT.
Wygląda w porządku, teraz sprawdzamy układ partycji:
Command (? for help): p
Disk /dev/sda: 156299375 sectors, 74.5 GiB
Logical sector size: 512 bytes
Disk identifier (GUID): 526AC8D7-EEC8-4BA3-B654-31EF40528AF2
Partition table holds up to 128 entries
First usable sector is 34, last usable sector is 156299341
Partitions will be aligned on 2048-sector boundaries
Total free space is 8236 sectors (4.0 MiB)
Number Start (sector) End (sector) Size Code Name
1 2048 61442047 29.3 GiB 0700 Microsoft basic data
2 61442048 83970047 10.7 GiB 8300 Linux filesystem
3 83970048 86067199 1024.0 MiB 8200 Linux swap
5 86069248 106549247 9.8 GiB 8300 Linux filesystem
6 106551296 127031295 9.8 GiB 8300 Linux filesystem
7 127033344 156299263 14.0 GiB 8300 Linux filesystem
Czyli bez problemu proces się zakończył. Rzućmy jeszcze okiem na to jak wygląda struktura partycji w parted:
# parted /dev/sda
GNU Parted 2.3
Using /dev/sda
Welcome to GNU Parted! Type 'help' to view a list of commands.
(parted) print free
Model: ATA WDC WD800JB-00JJ (scsi)
Disk /dev/sda: 80,0GB
Sector size (logical/physical): 512B/512B
Partition Table: gpt
Number Start End Size File system Name Flags
17,4kB 1049kB 1031kB Free Space
1 1049kB 31,5GB 31,5GB ntfs Microsoft basic data msftdata
2 31,5GB 43,0GB 11,5GB Linux filesystem
3 43,0GB 44,1GB 1074MB linux-swap(v1) Linux swap
44,1GB 44,1GB 1049kB Free Space
5 44,1GB 54,6GB 10,5GB Linux filesystem
54,6GB 54,6GB 1049kB Free Space
6 54,6GB 65,0GB 10,5GB ext4 Linux filesystem
65,0GB 65,0GB 1049kB Free Space
7 65,0GB 80,0GB 15,0GB Linux filesystem
80,0GB 80,0GB 39,9kB Free Space
No tu już widać trochę różnic. Przede wszystkim są dziury. W miejscu gdzie zaczynały się dyski na rozszerzonej partycji obecnie jest trochę wolnego miejsca. Dla lepszego rozeznania możemy ustawić jako jednostkę sektory:
(parted) unit s
(parted) print free
Model: ATA WDC WD800JB-00JJ (scsi)
Disk /dev/sda: 156299375s
Sector size (logical/physical): 512B/512B
Partition Table: gpt
Number Start End Size File system Name Flags
34s 2047s 2014s Free Space
1 2048s 61442047s 61440000s ntfs Microsoft basic data msftdata
2 61442048s 83970047s 22528000s Linux filesystem
3 83970048s 86067199s 2097152s linux-swap(v1) Linux swap
86067200s 86069247s 2048s Free Space
5 86069248s 106549247s 20480000s Linux filesystem
106549248s 106551295s 2048s Free Space
6 106551296s 127031295s 20480000s ext4 Linux filesystem legacy_boot
127031296s 127033343s 2048s Free Space
7 127033344s 156299263s 29265920s Linux filesystem
156299264s 156299341s 78s Free Space
albo bajty:
(parted) unit b
(parted) print free
Model: ATA WDC WD800JB-00JJ (scsi)
Disk /dev/sda: 80025280000B
Sector size (logical/physical): 512B/512B
Partition Table: gpt
Number Start End Size File system Name Flags
17408B 1048575B 1031168B Free Space
1 1048576B 31458328575B 31457280000B ntfs Microsoft basic data msftdata
2 31458328576B 42992664575B 11534336000B Linux filesystem
3 42992664576B 44066406399B 1073741824B linux-swap(v1) Linux swap
44066406400B 44067454975B 1048576B Free Space
5 44067454976B 54553214975B 10485760000B Linux filesystem
54553214976B 54554263551B 1048576B Free Space
6 54554263552B 65040023551B 10485760000B ext4 Linux filesystem legacy_boot
65040023552B 65041072127B 1048576B Free Space
7 65041072128B 80025223167B 14984151040B Linux filesystem
80025223168B 80025263103B 39936B Free Space
Jak widać jest to 2048 sektorów albo, jak kto woli, 2048*512=1048576 bajtów i jest to wynikiem równania do 1MiB. Można te przestrzenie pokasować np. w gparted przez rozszerzenie partycji i rozciągnięcie systemu plików, jeśli to komuś przeszkadza. Można też zauważyć, że ilość wolnego miejsca na początku i końcu dysku uległa zmianie.
Sprawdźmy zatem czy zaszyfrowany linux wciąż gdzieś tam jest na dysku:
# cryptsetup luksOpen /dev/sda2 sda2_crypt
Hasło dla /dev/sda2:
# mount /dev/mapper/sda2_crypt /mnt
# ls -al /mnt
razem 136
drwxr-xr-x 28 root root 4096 sty 3 05:55 .
drwxr-xr-x 31 root root 260 sty 4 12:15 ..
drwxr-xr-x 2 root root 4096 sty 4 07:55 bin
drwxr-xr-x 2 root root 4096 sty 4 10:16 boot
lrwxrwxrwx 1 root root 11 lis 11 01:00 cdrom -> media/cdrom
drwxr-xr-x 10 root root 4096 gru 30 10:24 cgroup
drwx------ 3 root root 4096 lis 11 02:16 .config
drwxr-xr-x 2 root root 4096 sty 4 09:52 dev
drwxr-xr-x 144 root root 12288 sty 4 10:48 etc
drwxr-xr-x 5 root root 4096 gru 21 09:32 home
drwxrwxrwt 2 root root 4096 lis 11 02:54 .ICE-unix
drwxr-xr-x 18 root root 4096 gru 30 03:04 lib
drwxr-xr-x 2 root root 4096 gru 6 19:57 lib64
drwx------ 2 root root 16384 lis 11 00:11 lost+found
drwxr-xr-x 15 root root 4096 sty 4 10:14 media
drwxr-xr-x 2 root root 4096 sty 4 09:52 mnt
drwxr-xr-x 9 root root 4096 gru 27 08:08 opt
dr-xr-xr-x 2 root root 4096 sty 4 07:59 proc
drwx------ 2 root root 4096 lis 11 02:54 pulse-PKdhtXMmr18n
drwxr-xr-x 23 root root 4096 sty 4 10:27 root
drwxr-xr-x 4 root root 4096 sty 4 11:00 run
drwxr-xr-x 2 root root 12288 sty 4 07:55 sbin
drwxr-xr-x 2 root root 4096 lis 11 00:18 srv
dr-xr-xr-x 2 root root 4096 sty 4 07:59 sys
drwxrwxrwt 9 root root 4096 sty 4 11:03 tmp
drwxr-xr-x 2 root root 4096 lis 11 02:52 tmp_ram
drwxr-xr-x 11 root root 4096 lis 11 07:16 usr
drwxr-xr-x 13 root root 4096 lis 12 20:52 var
drwxrwxrwt 2 root root 4096 lis 11 02:58 .X11-unix
Wygląda na to, że przetrwał proces konwersji i nic mu nie jest. Ale to nie jest koniec naszej pracy. Trzeba jeszcze wgrać bootloader, w tym przypadku będzie to extlinux. Tworzymy zatem środowisko pod chroot i wchodzimy do systemu:
# mount -o bind /dev /mnt/dev/
# mount -o bind /dev/pts /mnt/dev/pts
# mount -o bind /sys /mnt/sys
# mount -o bind /proc /mnt/proc
# chroot /mnt
W sgdisk ustawiamy bit drugi na partycję, która zawiera katalog /boot. W moim przypadku jest to partycja 6:
# sgdisk /dev/sda --attributes=6:set:2
The operation has completed successfully.
# sgdisk /dev/sda --attributes=6:show
6:2:1 (legacy BIOS bootable)
Jeśli ktoś jest ciekawe jakie bity można ustawić, poniżej rozpiska:
# sgdisk /dev/sda --attributes=list
0: system partition
1: hide from EFI
2: legacy BIOS bootable
60: read-only
62: hidden
63: do not automount
Wgrywamy teraz MBR i VBR, tak samo jak w przypadku tablicy partycji MBR:
# dd bs=440 conv=notrunc count=1 if=/usr/lib/syslinux/gptmbr.bin of=/dev/sda
# extlinux --install /boot/extlinux/
/boot/extlinux/ is device /dev/sda6
Drugie polecenie powinno utworzyć plik ldlinux.sys w katalogu /boot/extlinux .
Po tym zabiegu można rebootnąć pc. U mnie system wystartował. Jedyne co to winxp się nie chciał odpalić i to trzeba mieć na uwadze -- winxp nie wystartuje jeśli dysk ma tablicę partycji GPT.
W drugą stronę, czyli GPT -> MBR
Konwertowanie partycji z GPT na MBR jest trochę trudniejsze. Przede wszystkim potrzebne są dziury dla rozszerzonych partycji, te same, które postały przy konwersji z MBR na GPT. Jeśli ich nie mamy na dysku, prawdopodobnie nie da rady przenieść niektórych partycji z GPT na MBR. Można je zawsze stworzyć. Trzeba pamiętać by miały 2048 sektorów. Mi dziury zostały, także wykorzystam już to co jest. Odpalamy gdiska:
# gdisk /dev/sda
GPT fdisk (gdisk) version 0.8.8
Partition table scan:
MBR: protective
BSD: not present
APM: not present
GPT: present
Found valid GPT with protective MBR; using GPT.
Command (? for help): r
Recovery/transformation command (? for help): g
Warning: 0xEE partition doesn't start on sector 1. This can cause problems
in some OSes.
I to w zasadzie tyle roboty. Podejrzymy jeszcze jak będzie wyglądał nowy układ partycji:
MBR command (? for help): p
** NOTE: Partition numbers do NOT indicate final primary/logical status,
** unlike in most MBR partitioning tools!
** Extended partitions are not displayed, but will be generated as required.
Disk size is 156299375 sectors (74.5 GiB)
MBR disk identifier: 0x00000000
MBR partitions:
Can Be Can Be
Number Boot Start Sector End Sector Status Logical Primary Code
1 2048 61442047 primary Y 0x07
2 61442048 83970047 primary Y 0x83
3 83970048 86067199 primary Y 0x82
5 86069248 106549247 logical Y 0x83
6 106551296 127031295 logical Y 0x83
7 127033344 156299263 logical Y 0x83
Jak można zauważyć są 3 podstawowe i 3 logiczne na rozszerzonej partycji. Czyli wracamy do pozycji wyjściowej. Jeśli jednak by się zdarzyło, że układ nie odpowiada naszym oczekiwaniom, możemy go zmienić. Poniżej jest lista opcji, z których możemy skorzystać:
a toggle the active/boot flag
c recompute all CHS values
l set partition as logical
o omit partition
p print the MBR partition table
q quit without saving changes
r set partition as primary
s sort MBR partitions
t change partition type code
w write the MBR partition table to disk and exit
l - ustawia logiczną partycję, r - ustawia podstawową partycję, o - nie doda partycji do tablicy MBR.
Ja potrzebowałem tylko ustawić flagę boot na 6 partycję:
MBR command (? for help): a
Toggle active flag for partition: 6
MBR command (? for help): p
Disk size is 156299375 sectors (74.5 GiB)
MBR disk identifier: 0x00000000
MBR partitions:
Can Be Can Be
Number Boot Start Sector End Sector Status Logical Primary Code
1 2048 61442047 primary Y 0x07
2 61442048 83970047 primary Y 0x83
3 83970048 86067199 primary Y 0x82
5 86069248 106549247 logical Y 0x83
6 * 106551296 127031295 logical Y 0x83
7 127033344 156299263 logical Y 0x83
MBR command (? for help): w
Converted 6 partitions. Finalize and exit? (Y/N): y
GPT data structures destroyed! You may now partition the disk using fdisk or
other utilities.
Warning: 0xEE partition doesn't start on sector 1. This can cause problems
in some OSes.
U mnie po przekonwertowaniu z GPT na MBR trzeba było zresetować pc by zobaczyć zmiany. Poniżej się prezentuje dysk, po powrocie do tablicy partycji MBR:
# parted /dev/sda
GNU Parted 2.3
Using /dev/sda
Welcome to GNU Parted! Type 'help' to view a list of commands.
(parted) print free
Model: ATA WDC WD800JB-00JJ (scsi)
Disk /dev/sda: 80,0GB
Sector size (logical/physical): 512B/512B
Partition Table: msdos
Number Start End Size Type File system Flags
32,3kB 1049kB 1016kB Free Space
1 1049kB 31,5GB 31,5GB primary ntfs
2 31,5GB 43,0GB 11,5GB primary
3 43,0GB 44,1GB 1074MB primary linux-swap(v1)
44,1GB 44,1GB 1048kB Free Space
4 44,1GB 80,0GB 36,0GB extended lba
5 44,1GB 54,6GB 10,5GB logical
6 54,6GB 65,0GB 10,5GB logical ext4 boot
7 65,0GB 80,0GB 15,0GB logical
80,0GB 80,0GB 56,8kB Free Space
Jedyna różnica to wolne 2047 sektorów przed rozszerzoną partycją. Normalnie, to wolne miejsce by było przed pierwszym dyskiem na rozszerzonej partycji. Nie wpływa to jednak na nic.
By system się zabootował prawidłowo, trzeba jeszcze zreinstalować extlinuxa, tworzymy zatem środowisko pod chroot:
# cryptsetup luksOpen /dev/sda2 sda2_crypt
Hasło dla /dev/sda2:
# mount /dev/mapper/sda2_crypt /mnt
# mount /dev/sda6 /mnt/boot/
# mount -o bind /dev/ /mnt/dev/
# mount -o bind /dev/pts /mnt/dev/pts
# mount -o bind /proc /mnt/proc
# mount -o bind /sys /mnt/sys
# chroot /mnt/
I wgrywamy MBR i VBR:
# dd bs=440 conv=notrunc count=1 if=/usr/lib/syslinux/mbr.bin of=/dev/sda
1+0 przeczytanych recordów
1+0 zapisanych recordów
skopiowane 440 bajtów (440 B), 0,0300519 s, 14,6 kB/s
# extlinux --install /boot/extlinux/
/boot/extlinux/ is device /dev/sda6
Po resecie nawet windows się odpalił. Znaczy, że wszystko wróciło do normy.